From e2ed31dd0112dc3bede53ceef9b957d2810e141e Mon Sep 17 00:00:00 2001 From: joonhoekim <26rote@gmail.com> Date: Wed, 27 Aug 2025 08:24:58 +0000 Subject: (김준회) 임시 관리자 페이지 - EDP 데이터 수동 관리 추가 및 세션검증 추가 MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../admin/edp/components/project-selector.tsx | 258 +++++++++++++++++++++ 1 file changed, 258 insertions(+) create mode 100644 app/[lng]/admin/edp/components/project-selector.tsx (limited to 'app/[lng]/admin/edp/components/project-selector.tsx') diff --git a/app/[lng]/admin/edp/components/project-selector.tsx b/app/[lng]/admin/edp/components/project-selector.tsx new file mode 100644 index 00000000..68895cd1 --- /dev/null +++ b/app/[lng]/admin/edp/components/project-selector.tsx @@ -0,0 +1,258 @@ +'use client' + +import { useState, useEffect } from 'react' +import { Button } from '@/components/ui/button' +import { Dialog, DialogContent, DialogHeader, DialogTitle, DialogTrigger } from '@/components/ui/dialog' +import { Input } from '@/components/ui/input' +import { Badge } from '@/components/ui/badge' +import { Search, Check } from 'lucide-react' +import { + ColumnDef, + flexRender, + getCoreRowModel, + getFilteredRowModel, + getPaginationRowModel, + getSortedRowModel, + useReactTable, + SortingState, + ColumnFiltersState, + VisibilityState, + RowSelectionState, +} from '@tanstack/react-table' +import { + Table, + TableBody, + TableCell, + TableHead, + TableHeader, + TableRow, +} from '@/components/ui/table' +import { getProjects } from '../actions/data-actions' +import { toast } from 'sonner' + +interface Project { + id: number + code: string + name: string + type: string +} + +interface ProjectSelectorProps { + selectedProject?: Project + onProjectSelect: (project: Project) => void + disabled?: boolean +} + +export function ProjectSelector({ selectedProject, onProjectSelect, disabled }: ProjectSelectorProps) { + const [open, setOpen] = useState(false) + const [projects, setProjects] = useState([]) + const [loading, setLoading] = useState(false) + const [sorting, setSorting] = useState([]) + const [columnFilters, setColumnFilters] = useState([]) + const [columnVisibility, setColumnVisibility] = useState({}) + const [rowSelection, setRowSelection] = useState({}) + const [globalFilter, setGlobalFilter] = useState('') + + const columns: ColumnDef[] = [ + { + accessorKey: 'code', + header: '프로젝트 코드', + cell: ({ row }) => ( +
{row.getValue('code')}
+ ), + }, + { + accessorKey: 'name', + header: '프로젝트명', + cell: ({ row }) => ( +
{row.getValue('name')}
+ ), + }, + { + accessorKey: 'type', + header: '타입', + cell: ({ row }) => ( + {row.getValue('type')} + ), + }, + { + id: 'actions', + header: '선택', + cell: ({ row }) => ( + + ), + }, + ] + + const table = useReactTable({ + data: projects, + columns, + onSortingChange: setSorting, + onColumnFiltersChange: setColumnFilters, + onColumnVisibilityChange: setColumnVisibility, + onRowSelectionChange: setRowSelection, + onGlobalFilterChange: setGlobalFilter, + getCoreRowModel: getCoreRowModel(), + getPaginationRowModel: getPaginationRowModel(), + getSortedRowModel: getSortedRowModel(), + getFilteredRowModel: getFilteredRowModel(), + state: { + sorting, + columnFilters, + columnVisibility, + rowSelection, + globalFilter, + }, + }) + + const loadProjects = async () => { + setLoading(true) + try { + const result = await getProjects() + if (result.success) { + setProjects(result.data) + } else { + toast.error(result.error) + } + } catch (error) { + toast.error('프로젝트를 불러오는 중 오류가 발생했습니다.') + } finally { + setLoading(false) + } + } + + const handleProjectSelect = (project: Project) => { + onProjectSelect(project) + setOpen(false) + } + + useEffect(() => { + if (open && projects.length === 0) { + loadProjects() + } + }, [open]) + + return ( + + + + + + + 프로젝트 선택 + + +
+
+ + setGlobalFilter(e.target.value)} + className="flex-1" + /> +
+ + {loading ? ( +
+
프로젝트를 불러오는 중...
+
+ ) : ( +
+ + + {table.getHeaderGroups().map((headerGroup) => ( + + {headerGroup.headers.map((header) => ( + + {header.isPlaceholder + ? null + : flexRender( + header.column.columnDef.header, + header.getContext() + )} + + ))} + + ))} + + + {table.getRowModel().rows?.length ? ( + table.getRowModel().rows.map((row) => ( + handleProjectSelect(row.original)} + > + {row.getVisibleCells().map((cell) => ( + + {flexRender( + cell.column.columnDef.cell, + cell.getContext() + )} + + ))} + + )) + ) : ( + + + 검색 결과가 없습니다. + + + )} + +
+
+ )} + +
+
+ 총 {table.getFilteredRowModel().rows.length}개 프로젝트 +
+
+ +
+ {table.getState().pagination.pageIndex + 1} / {table.getPageCount()} +
+ +
+
+
+
+
+ ) +} -- cgit v1.2.3